package com.zpdhy.system.shiro;

import com.zpdhy.system.model.SysPermission;
import com.zpdhy.system.model.SysRole;
import com.zpdhy.system.model.UserInfo;
import com.zpdhy.system.service.UserInfoService;
import org.apache.shiro.SecurityUtils;
import org.apache.shiro.authc.AuthenticationException;
import org.apache.shiro.authc.AuthenticationInfo;
import org.apache.shiro.authc.AuthenticationToken;
import org.apache.shiro.authc.SimpleAuthenticationInfo;
import org.apache.shiro.authz.AuthorizationInfo;
import org.apache.shiro.authz.SimpleAuthorizationInfo;
import org.apache.shiro.realm.AuthorizingRealm;
import org.apache.shiro.subject.PrincipalCollection;
import org.apache.shiro.subject.SimplePrincipalCollection;
import org.apache.shiro.subject.Subject;
import org.apache.shiro.util.ByteSource;

import javax.annotation.Resource;
import java.util.HashSet;
import java.util.List;
import java.util.Set;

/**
 * 身份校验核心类;
 * Created by kedong on 2016/12/30.
 */
public class ShiroRealm extends AuthorizingRealm {

    @Resource
    private UserInfoService userInfoService;

    /**
     * 此方法调用  hasRole,hasPermission的时候才会进行回调.
     * <p>
     * 权限信息.(授权):
     * 1、如果用户正常退出，缓存自动清空；
     * 2、如果用户非正常退出，缓存自动清空；
     * 3、如果我们修改了用户的权限，而用户不退出系统，修改的权限无法立即生效。
     * （需要手动编程进行实现；放在service进行调用）
     * 在权限修改后调用realm中的方法，realm已经由spring管理，所以从spring中获取realm实例，
     * 调用clearCached方法；
     * :Authorization 是授权访问控制，用于对用户进行的操作授权，证明该用户是否允许进行当前操作，如访问某个链接，某个资源文件等。
     *
     * @param principalCollection
     * @return
     */
    @Override
    protected AuthorizationInfo doGetAuthorizationInfo(PrincipalCollection principalCollection) {
        System.out.println("-------------------ShiroRealm.doGetAuthorizationInfo()");
        /*
        * 当没有使用缓存的时候，不断刷新页面的话，这个代码会不断执行，
        * 当其实没有必要每次都重新设置权限信息，所以我们需要放到缓存中进行管理；
        * 当放到缓存中时，这样的话，doGetAuthorizationInfo就只会执行一次了，
        * 缓存过期之后会再次执行。
        */
        SimpleAuthorizationInfo authorizationInfo = new SimpleAuthorizationInfo();
        String username = (String) principalCollection.getPrimaryPrincipal();

        //实际项目中，这里可以根据实际情况做缓存，如果不做，Shiro自己也是有时间间隔机制，2分钟内不会重复执行该方法
        UserInfo userInfo = userInfoService.findByUsername(username);

        /**
         权限单个添加;
         或者按下面这样添加
         添加一个角色,不是配置意义上的添加,而是证明该用户拥有admin角色
         authorizationInfo.addRole("admin");
         添加权限
         authorizationInfo.addStringPermission("userInfo:query");


         在认证成功之后返回.
         设置角色信息.
         支持 Set集合,
         用户的角色对应的所有权限，如果只使用角色定义访问权限，下面的四行可以不要
         List<Role> roleList=user.getRoleList();
         for (Role role : roleList) {
         info.addStringPermissions(role.getPermissionsName());
         }
         */

        for (SysRole role : userInfo.getRoleList()) {
            // 设置角色
            authorizationInfo.addRole(role.getRole());
            for (SysPermission p : role.getPermissions()) {
                // 设置权限
                authorizationInfo.addStringPermission(p.getPermission());
            }
        }

        //设置权限信息.
//     authorizationInfo.setStringPermissions(getStringPermissions(userInfo.getRoleList()));

        return authorizationInfo;

    }

    /**
     * 认证信息.(身份验证)
     * Authentication 是用来验证用户身份
     *
     * @param authenticationToken
     * @return
     * @throws AuthenticationException
     */
    @Override
    protected AuthenticationInfo doGetAuthenticationInfo(AuthenticationToken authenticationToken) throws AuthenticationException {
        System.out.println("-------------------ShiroRealm.doGetAuthenticationInfo()");

        //获取用户的输入的账号.
        String username = (String) authenticationToken.getPrincipal();
        System.out.println(username);

        /**
         * 通过username从数据库中查找 User对象，如果找到，没找到.
         * 实际项目中，这里可以根据实际情况做缓存，如果不做，Shiro自己也是有时间间隔机制，2分钟内不会重复执行该方法
         */
        UserInfo userInfo = userInfoService.findByUsername(username);
        System.out.println("----->> " + userInfo);
        if (userInfo == null) {
            return null;
        }

        /*
        * 获取权限信息:这里没有进行实现，
        * 请自行根据UserInfo,Role,Permission进行实现；
        * 获取之后可以在前端for循环显示所有链接;
        */
        //userInfo.setPermissions(userService.findPermissions(user));

        //账号判断;
        //加密方式;
        //交给AuthenticatingRealm使用CredentialsMatcher进行密码匹配，如果觉得人家的不好可以自定义实现
        SimpleAuthenticationInfo authenticationInfo = new SimpleAuthenticationInfo(
                username,
                userInfo.getPassword(),
                ByteSource.Util.bytes(userInfo.getUserName() + userInfo.getSalt()),
                getName());

        //明文: 若存在，将此用户存放到登录认证info中，无需自己做密码对比，Shiro会为我们进行密码对比校验
//      SimpleAuthenticationInfo authenticationInfo = new SimpleAuthenticationInfo(
//           userInfo, //用户名
//           userInfo.getPassword(), //密码
//             getName()  //realm name
//      );

        return authenticationInfo;
    }

    /**
     * 将权限对象中的权限code取出.
     *
     * @param permissions
     * @return
     */
    public Set<String> getStringPermissions(List<SysPermission> permissions) {
        Set<String> stringPermissions = new HashSet<String>();
        if (permissions != null) {
            for (SysPermission p : permissions) {
                stringPermissions.add(p.getPermission());
            }
        }
        return stringPermissions;
    }

}
